If you want to see how the examples work underneath, see the markdown source
Most markdown renderers allow you to mix html and markdown.
We can take advantage of this to add html <scripts>
elements that will get executed in the browser.
To execute code blocks and show the results below, add this script at the end of your markdown file:
<!-- end of markdown file -->
<script>
...document.querySelectorAll('pre > code')].forEach(code => {
[const result = eval(code.innerText);
// show result in a div after the code block
const resultElement = document.createElement('div');
.innerText = result;
resultElement.after(resultElement);
code
})</script>
new Date();
I am trying to make some simulations in canvas because I want to see
the results in real time.
This script expects a function in the code blocks and passes the canvas
so you can draw in them.
<!-- end of markdown file -->
<script>
...document.querySelectorAll('pre > code')].forEach(code => {
[// Add a canvas after the code block
const canvas = document.createElement("canvas");
.parentElement.after(canvas);
codeconst ctx = canvas.getContext('2d');
// Run the code
// Note: Code blocks must return a function so they can receive the canvas context and element as parameters
const fn = eval(code.innerText);
fn(ctx, canvas);
})</script>
, canvas) => {
(ctx.width = 200;
canvas.height = 200;
canvas.style.backgroundColor = 'black';
canvas
let x = Math.random() * canvas.width;
let y = Math.random() * canvas.height;
let dx = 4 * Math.random() * 2 - 1;
let dy = 4 * Math.random() * 2 - 1;
function animate() {
.fillStyle = 'rgba(0, 0, 0, 0.1)';
ctx.fillRect(0, 0, canvas.width, canvas.height);
ctx
.fillStyle = 'white';
ctx.beginPath();
ctx.arc(x, y, 5, 0, Math.PI * 2);
ctx.fill();
ctx
+= dx;
x += dy;
y
if (x < 0 || x > canvas.width) dx = -dx;
if (y < 0 || y > canvas.height) dy = -dy;
requestAnimationFrame(animate);
}
animate();
}
We can mark the code blocks we want to run with a comment like
// run-on-canvas
at the top of the block.
And filter them with this line:
.filter(code => code.children[0].innerText === '// run-on-canvas')
<!-- end of markdown file -->
<script>
...document.querySelectorAll('pre > code')]
[.filter(code => code.children[0].innerText === '// run-on-canvas')
.forEach(code => {
// Add a canvas after the code block
const pre = code.parentElement;
const canvas = document.createElement("canvas");
.after(canvas);
preconst ctx = canvas.getContext('2d');
// Run the code
const fn = eval(code.innerText);
fn(ctx, canvas);
// Remove tag
// code.childNodes[0].remove(); // remove tag
// code.childNodes[0].remove(); // remove extra line
})</script>
You may have noticed that this script has code to remove the
// run-on-canvas
tag from the blocks, but I commented that
out so you can see which blocks were tagged in the examples below.
// run-on-canvas
=> {
ctx .fillStyle = 'blue';
ctx.fillRect(10, 10, 100, 50);
ctx }
// this block does not run
=> {
ctx .fillStyle = 'red';
ctx.fillRect(10, 10, 100, 50);
ctx }
This last code block does not run because it lacks the
// run-on-canvas
tag.
Scripts in this post are not executed “as is”:
You can see the markdown source of this post here: markdown source